home *** CD-ROM | disk | FTP | other *** search
/ Chip 2007 January, February, March & April / Chip-Cover-CD-2007-02.iso / Pakiet internetowy / Przegladarki internetowe / Mozilla Seamonkey 1.0.5 pl / seamonkey-1.0.5.pl-PL.win32.installer.exe / VENKMAN.XPI / bin / chrome / venkman.jar / content / venkman / venkman-debugger.js < prev    next >
Encoding:
Text File  |  2004-04-18  |  55.0 KB  |  2,046 lines

  1. /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
  2.  *
  3.  * ***** BEGIN LICENSE BLOCK *****
  4.  * Version: MPL 1.1/GPL 2.0/LGPL 2.1
  5.  *
  6.  * The contents of this file are subject to the Mozilla Public License Version
  7.  * 1.1 (the "License"); you may not use this file except in compliance with
  8.  * the License. You may obtain a copy of the License at
  9.  * http://www.mozilla.org/MPL/
  10.  *
  11.  * Software distributed under the License is distributed on an "AS IS" basis,
  12.  * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
  13.  * for the specific language governing rights and limitations under the
  14.  * License.
  15.  *
  16.  * The Original Code is The JavaScript Debugger.
  17.  *
  18.  * The Initial Developer of the Original Code is
  19.  * Netscape Communications Corporation.
  20.  * Portions created by the Initial Developer are Copyright (C) 1998
  21.  * the Initial Developer. All Rights Reserved.
  22.  *
  23.  * Contributor(s):
  24.  *   Robert Ginda, <rginda@netscape.com>, original author
  25.  *
  26.  * Alternatively, the contents of this file may be used under the terms of
  27.  * either the GNU General Public License Version 2 or later (the "GPL"), or
  28.  * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
  29.  * in which case the provisions of the GPL or the LGPL are applicable instead
  30.  * of those above. If you wish to allow use of your version of this file only
  31.  * under the terms of either the GPL or the LGPL, and not to allow others to
  32.  * use your version of this file under the terms of the MPL, indicate your
  33.  * decision by deleting the provisions above and replace them with the notice
  34.  * and other provisions required by the GPL or the LGPL. If you do not delete
  35.  * the provisions above, a recipient may use your version of this file under
  36.  * the terms of any one of the MPL, the GPL or the LGPL.
  37.  *
  38.  * ***** END LICENSE BLOCK ***** */
  39.  
  40. const JSD_CTRID           = "@mozilla.org/js/jsd/debugger-service;1";
  41. const jsdIDebuggerService = Components.interfaces.jsdIDebuggerService;
  42. const jsdIExecutionHook   = Components.interfaces.jsdIExecutionHook;
  43. const jsdIErrorHook       = Components.interfaces.jsdIErrorHook;
  44. const jsdICallHook        = Components.interfaces.jsdICallHook;
  45. const jsdIValue           = Components.interfaces.jsdIValue;
  46. const jsdIProperty        = Components.interfaces.jsdIProperty;
  47. const jsdIScript          = Components.interfaces.jsdIScript;
  48. const jsdIStackFrame      = Components.interfaces.jsdIStackFrame;
  49.  
  50. const TYPE_VOID     = jsdIValue.TYPE_VOID;
  51. const TYPE_NULL     = jsdIValue.TYPE_NULL;
  52. const TYPE_BOOLEAN  = jsdIValue.TYPE_BOOLEAN;
  53. const TYPE_INT      = jsdIValue.TYPE_INT;
  54. const TYPE_DOUBLE   = jsdIValue.TYPE_DOUBLE;
  55. const TYPE_STRING   = jsdIValue.TYPE_STRING;
  56. const TYPE_FUNCTION = jsdIValue.TYPE_FUNCTION;
  57. const TYPE_OBJECT   = jsdIValue.TYPE_OBJECT;
  58.  
  59. const PROP_ENUMERATE = jsdIProperty.FLAG_ENUMERATE;
  60. const PROP_READONLY  = jsdIProperty.FLAG_READONLY;
  61. const PROP_PERMANENT = jsdIProperty.FLAG_PERMANENT;
  62. const PROP_ALIAS     = jsdIProperty.FLAG_ALIAS;
  63. const PROP_ARGUMENT  = jsdIProperty.FLAG_ARGUMENT;
  64. const PROP_VARIABLE  = jsdIProperty.FLAG_VARIABLE;
  65. const PROP_EXCEPTION = jsdIProperty.FLAG_EXCEPTION;
  66. const PROP_ERROR     = jsdIProperty.FLAG_ERROR;
  67. const PROP_HINTED    = jsdIProperty.FLAG_HINTED;
  68.  
  69. const SCRIPT_NODEBUG   = jsdIScript.FLAG_DEBUG;
  70. const SCRIPT_NOPROFILE = jsdIScript.FLAG_PROFILE;
  71.  
  72. const COLLECT_PROFILE_DATA  = jsdIDebuggerService.COLLECT_PROFILE_DATA;
  73.  
  74. const PCMAP_SOURCETEXT    = jsdIScript.PCMAP_SOURCETEXT;
  75. const PCMAP_PRETTYPRINT   = jsdIScript.PCMAP_PRETTYPRINT;
  76.  
  77. const RETURN_CONTINUE   = jsdIExecutionHook.RETURN_CONTINUE;
  78. const RETURN_CONT_THROW = jsdIExecutionHook.RETURN_CONTINUE_THROW;
  79. const RETURN_VALUE      = jsdIExecutionHook.RETURN_RET_WITH_VAL;
  80. const RETURN_THROW      = jsdIExecutionHook.RETURN_THROW_WITH_VAL;
  81.  
  82. const FTYPE_STD     = 0;
  83. const FTYPE_SUMMARY = 1;
  84. const FTYPE_ARRAY   = 2;
  85.  
  86. const BREAKPOINT_STOPNEVER   = 0;
  87. const BREAKPOINT_STOPALWAYS  = 1;
  88. const BREAKPOINT_STOPTRUE    = 2;
  89. const BREAKPOINT_EARLYRETURN = 3;
  90.  
  91. var $ = new Array(); /* array to store results from evals in debug frames */
  92.  
  93. function compareVersion(maj, min)
  94. {
  95.     if (console.jsds.implementationMajor < maj)
  96.         return -1;
  97.     
  98.     if (console.jsds.implementationMajor > maj)
  99.         return 1;
  100.  
  101.     if (console.jsds.implementationMinor < min)
  102.         return -1;
  103.     
  104.     if (console.jsds.implementationMinor > min)
  105.         return 1;
  106.  
  107.     return 0;
  108. }
  109.  
  110. function initDebugger()
  111. {   
  112.     dd ("initDebugger {");
  113.  
  114.     console.instanceSequence = 0;
  115.     console._continueCodeStack = new Array(); /* top of stack is the default  */
  116.                                               /* return code for the most     */
  117.                                               /* recent debugTrap().          */
  118.     console.scriptWrappers = new Object();
  119.     console.scriptManagers = new Object();
  120.     console.breaks  = new Object();
  121.     console.fbreaks = new Object();
  122.     console.sbreaks = new Object();
  123.  
  124.     /* create the debugger instance */
  125.     if (!(JSD_CTRID in Components.classes))
  126.         throw new BadMojo (ERR_NO_DEBUGGER);
  127.     
  128.     console.jsds = 
  129.         Components.classes[JSD_CTRID].getService(jsdIDebuggerService);
  130.     console.jsds.on();
  131.  
  132.     if (compareVersion(1, 2) >= 0)
  133.         console.jsds.flags = jsdIDebuggerService.DISABLE_OBJECT_TRACE;
  134.  
  135.     console.executionHook = { onExecute: jsdExecutionHook };
  136.     console.errorHook     = { onError: jsdErrorHook };
  137.     console.callHook      = { onCall: jsdCallHook };
  138.     
  139.     console.jsdConsole = console.jsds.wrapValue(console);    
  140.  
  141.     dispatch ("tmode", {mode: console.prefs["lastThrowMode"]});
  142.     dispatch ("emode", {mode: console.prefs["lastErrorMode"]});
  143.     
  144.     var enumer = { enumerateScript: console.scriptHook.onScriptCreated };
  145.     console.jsds.scriptHook = console.scriptHook;
  146.     console.jsds.enumerateScripts(enumer);
  147.  
  148.     console.jsds.breakpointHook = console.executionHook;
  149.     console.jsds.debuggerHook   = console.executionHook;
  150.     console.jsds.debugHook      = console.executionHook;
  151.     console.jsds.errorHook      = console.errorHook;
  152.  
  153.     console.jsds.flags = jsdIDebuggerService.ENABLE_NATIVE_FRAMES;
  154.  
  155.     dd ("} initDebugger");
  156. }
  157.  
  158. function detachDebugger()
  159. {
  160.     if ("frames" in console)
  161.         console.jsds.exitNestedEventLoop();
  162.  
  163.     var b;
  164.     for (b in console.breaks)
  165.         console.breaks[b].clearBreakpoint();
  166.     for (b in console.fbreaks)
  167.         console.fbreaks[b].clearFutureBreakpoint();
  168.  
  169.     console.jsds.topLevelHook = null;
  170.     console.jsds.functionHook = null;
  171.     console.jsds.breakpointHook = null;
  172.     console.jsds.debuggerHook = null;
  173.     console.jsds.errorHook = null;
  174.     console.jsds.scriptHook = null;
  175.     console.jsds.interruptHook = null;
  176.     console.jsds.clearAllBreakpoints();
  177.  
  178.     console.jsds.GC();
  179.  
  180.     if (!console.jsds.initAtStartup)
  181.         console.jsds.off();
  182. }
  183.  
  184. console.scriptHook = new Object();
  185.  
  186. console.scriptHook.onScriptCreated =
  187. function sh_created (jsdScript)
  188. {
  189.     try
  190.     {
  191.         jsdScriptCreated(jsdScript);
  192.     }
  193.     catch (ex)
  194.     {
  195.         dd ("caught " + dumpObjectTree(ex) + " while creating script.");
  196.     }
  197. }
  198.  
  199. console.scriptHook.onScriptDestroyed =
  200. function sh_destroyed (jsdScript)
  201. {
  202.     try
  203.     {
  204.         jsdScriptDestroyed(jsdScript);
  205.     }
  206.     catch (ex)
  207.     {
  208.         dd ("caught " + dumpObjectTree(ex) + " while destroying script.");
  209.     }
  210. }
  211.     
  212. function jsdScriptCreated (jsdScript)
  213. {
  214.     var url = jsdScript.fileName;
  215.     var manager;
  216.     
  217.     if (!(url in console.scriptManagers))
  218.     {
  219.         manager = console.scriptManagers[url] = new ScriptManager(url);
  220.         //dispatchCommand (console.coManagerCreated, { scriptManager: manager });
  221.     }
  222.     else
  223.     {
  224.         manager = console.scriptManagers[url];
  225.     }
  226.     
  227.     manager.onScriptCreated(jsdScript);
  228. }
  229.  
  230. function jsdScriptDestroyed (jsdScript)
  231. {
  232.     if (!(jsdScript.tag in console.scriptWrappers))
  233.         return;
  234.     
  235.     var scriptWrapper = console.scriptWrappers[jsdScript.tag];
  236.     scriptWrapper.scriptManager.onScriptInvalidated(scriptWrapper);
  237.     
  238.     if (scriptWrapper.scriptManager.instances.length == 0 && 
  239.         scriptWrapper.scriptManager.transientCount == 0)
  240.     {
  241.         delete console.scriptManagers[scriptWrapper.scriptManager.url];
  242.         //dispatchCommand (console.coManagerDestroyed,
  243.         //                 { scriptManager: scriptWrapper.scriptManager });
  244.     }
  245. }
  246.  
  247. function jsdExecutionHook (frame, type, rv)
  248. {
  249.     dd ("execution hook: " + formatFrame(frame));
  250.     
  251.     var hookReturn = jsdIExecutionHook.RETURN_CONTINUE;
  252.  
  253.     if (!console.initialized)
  254.         return hookReturn;
  255.     
  256.  
  257.     if (!ASSERT(!("frames" in console),
  258.                 "Execution hook called while stopped") ||
  259.         frame.isNative ||
  260.         !ASSERT(frame.script, "Execution hook called with no script") ||
  261.         frame.script.fileName == MSG_VAL_CONSOLE ||
  262.         !ASSERT(!(frame.script.flags & SCRIPT_NODEBUG),
  263.                 "Stopped in a script marked as don't debug") ||
  264.         !ASSERT(isURLVenkman(frame.script.fileName) ||
  265.                 !isURLFiltered(frame.script.fileName),
  266.                 "stopped in a filtered URL"))
  267.     {
  268.         return hookReturn;
  269.     }
  270.  
  271.     var frames = new Array();
  272.     var prevFrame = frame;
  273.     var hasDisabledFrame = false;
  274.     
  275.     while (prevFrame)
  276.     {
  277.         frames.push(prevFrame);
  278.         prevFrame = prevFrame.callingFrame;
  279.     }
  280.  
  281.     var targetWindow = null;
  282.     var wasModal = false;
  283.     var cx;
  284.  
  285.     try
  286.     {
  287.         cx = frame.executionContext;
  288.     }
  289.     catch (ex)
  290.     {
  291.         dd ("no context");
  292.         cx = null;
  293.     }
  294.     
  295.     var targetWasEnabled = true;
  296.     var debuggerWasEnabled = console.baseWindow.enabled;
  297.     console.baseWindow.enabled = true;
  298.     
  299.     if (!ASSERT(cx, "no cx in execution hook"))
  300.         return hookReturn;
  301.     
  302.     var glob = cx.globalObject;
  303.     if (!ASSERT(glob, "no glob in execution hook"))
  304.         return hookReturn;
  305.     
  306.     console.targetWindow = getBaseWindowFromWindow(glob.getWrappedValue());
  307.     targetWasEnabled = console.targetWindow.enabled;
  308.     if (console.targetWindow != console.baseWindow)
  309.     {
  310.         cx.scriptsEnabled = false;
  311.         console.targetWindow.enabled = false;
  312.     }
  313.  
  314.     try
  315.     {
  316.         //dd ("debug trap " + formatFrame(frame));
  317.         hookReturn = debugTrap(frames, type, rv);
  318.         //dd ("debug trap returned " + hookReturn);
  319.     }
  320.     catch (ex)
  321.     {
  322.         display (MSG_ERR_INTERNAL_BPT, MT_ERROR);
  323.         display (formatException(ex), MT_ERROR);
  324.     }
  325.     
  326.  
  327.     
  328.     if (console.targetWindow && console.targetWindow != console.baseWindow)
  329.     {
  330.         console.targetWindow.enabled = targetWasEnabled;
  331.         cx.scriptsEnabled = true;
  332.     }
  333.     
  334.     console.baseWindow.enabled = debuggerWasEnabled;
  335.     delete console.frames;
  336.     delete console.targetWindow;
  337.     if ("__exitAfterContinue__" in console)
  338.         window.close();
  339.  
  340.     return hookReturn;
  341. }
  342.  
  343. function jsdCallHook (frame, type)
  344. {
  345.     if (!console.initialized)
  346.         return;
  347.     
  348.     if (type == jsdICallHook.TYPE_FUNCTION_CALL)
  349.     {
  350.         setStopState(false);
  351.         //dd ("Calling: " + frame.functionName);
  352.     }
  353.     else if (type == jsdICallHook.TYPE_FUNCTION_RETURN)
  354.     {
  355.         // we're called *before* the returning frame is popped from the
  356.         // stack, so we want our depth calculation to be off by one.
  357.         var depth = -1;
  358.         var prevFrame = frame;
  359.         
  360.         while (prevFrame)
  361.         {
  362.             depth++;
  363.             prevFrame = prevFrame.callingFrame;
  364.         }
  365.  
  366.         
  367.         //dd ("Returning: " + frame.functionName +
  368.         //    ", target depth: " + console._stepOverDepth +
  369.         //    ", current depth: " + depth);
  370.         
  371.         if (depth <= console._stepOverDepth)
  372.         {
  373.             //dd ("step over at target depth of " + depth);
  374.             setStopState(true);
  375.             console.jsds.functionHook = null;
  376.             delete console._stepOverDepth;
  377.         }
  378.     }
  379. }
  380.  
  381. function jsdErrorHook (message, fileName, line, pos, flags, exception)
  382. {
  383.     if (!console.initialized || isURLFiltered (fileName))
  384.         return true;
  385.     
  386.     try
  387.     {
  388.         var flagstr;
  389.         flagstr =
  390.             (flags && jsdIErrorHook.REPORT_EXCEPTION) ? "x" : "-";
  391.         flagstr +=
  392.             (flags && jsdIErrorHook.REPORT_STRICT) ? "s" : "-";
  393.         
  394.         //dd ("===\n" + message + "\n" + fileName + "@" + 
  395.         //    line + ":" + pos + "; " + flagstr);
  396.         var msn = (flags & jsdIErrorHook.REPORT_WARNING) ?
  397.             MSN_ERPT_WARN : MSN_ERPT_ERROR;
  398.  
  399.         if (console.errorMode != EMODE_IGNORE)
  400.             display (getMsg(msn, [message, flagstr, fileName,
  401.                                   line, pos]), MT_ETRACE);
  402.         
  403.         if (console.errorMode == EMODE_BREAK)
  404.             return false;
  405.         
  406.         return true;
  407.     }
  408.     catch (ex)
  409.     {
  410.         dd ("error in error hook: " + ex);
  411.     }
  412.     return true;
  413. }
  414.  
  415. function ScriptManager (url)
  416. {
  417.     this.url = url;
  418.     this.instances = new Array();
  419.     this.transients = new Object();
  420.     this.transientCount = 0;
  421.     this.disableTransients = isURLFiltered(url);
  422. }
  423.  
  424. ScriptManager.prototype.onScriptCreated =
  425. function smgr_created (jsdScript)
  426. {
  427.     var instance;
  428.  
  429.     if (!ASSERT(jsdScript.isValid, "invalid script created!"))
  430.         return;
  431.     
  432.     if (this.instances.length != 0)
  433.         instance = this.instances[this.instances.length - 1];
  434.  
  435.     if (!instance || (instance.isSealed && jsdScript.functionName))
  436.     {
  437.         //dd ("instance created for " + jsdScript.fileName);
  438.         instance = new ScriptInstance(this);
  439.         instance.sequence = console.instanceSequence++;
  440.         this.instances.push(instance);
  441.         dispatchCommand (console.coInstanceCreated,
  442.                          { scriptInstance: instance });
  443.     }
  444.  
  445.     if ("_lastScriptWrapper" in console)
  446.     {
  447.         if ((console._lastScriptWrapper.scriptManager != this ||
  448.              console._lastScriptWrapper.scriptInstance != instance) &&
  449.             console._lastScriptWrapper.scriptInstance.scriptCount &&
  450.             !console._lastScriptWrapper.scriptInstance.isSealed)
  451.         {
  452.             console._lastScriptWrapper.scriptInstance.seal();
  453.         }
  454.     }
  455.             
  456.     var scriptWrapper = new ScriptWrapper(jsdScript);
  457.     console._lastScriptWrapper = scriptWrapper;
  458.     scriptWrapper.scriptManager = this;
  459.     console.scriptWrappers[jsdScript.tag] = scriptWrapper;
  460.     scriptWrapper.scriptInstance = instance;
  461.  
  462.     if (!instance.isSealed)
  463.     {
  464.         //dd ("function created " + formatScript(jsdScript));
  465.         instance.onScriptCreated (scriptWrapper);
  466.     }
  467.     else
  468.     {
  469.         //dd ("transient created " + formatScript(jsdScript));
  470.         ++this.transientCount;
  471.         if (this.disableTransients)
  472.             jsdScript.flags |= SCRIPT_NODEBUG | SCRIPT_NOPROFILE;
  473.  
  474.         this.transients[jsdScript.tag] = scriptWrapper;
  475.         scriptWrapper.functionName = MSG_VAL_EVSCRIPT;
  476.         //dispatch ("hook-transient-script", { scriptWrapper: scriptWrapper });
  477.     }    
  478. }
  479.  
  480. ScriptManager.prototype.onScriptInvalidated =
  481. function smgr_invalidated (scriptWrapper)
  482. {
  483.     //dd ("script invalidated");
  484.     
  485.     delete console.scriptWrappers[scriptWrapper.tag];
  486.     if (scriptWrapper.tag in this.transients)
  487.     {
  488.         //dd ("transient destroyed " + formatScript(scriptWrapper.jsdScript));
  489.         --this.transientCount;
  490.         delete this.transients[scriptWrapper.tag];
  491.         //dispatch ("hook-script-invalidated", { scriptWrapper: scriptWrapper });
  492.     }
  493.     else
  494.     {
  495.         //dd ("function destroyed " + formatScript(scriptWrapper.jsdScript));
  496.         scriptWrapper.scriptInstance.onScriptInvalidated(scriptWrapper);
  497.         //dispatch ("hook-script-invalidated", { scriptWrapper: scriptWrapper });
  498.  
  499.         if (scriptWrapper.scriptInstance.scriptCount == 0)
  500.         {
  501.             var i = arrayIndexOf(this.instances, scriptWrapper.scriptInstance);
  502.             arrayRemoveAt(this.instances, i);
  503.             dispatchCommand (console.coInstanceDestroyed,
  504.                              { scriptInstance: scriptWrapper.scriptInstance });
  505.         }
  506.     }
  507. }    
  508.  
  509. ScriptManager.prototype.__defineGetter__ ("sourceText", smgr_sourcetext);
  510. function smgr_sourcetext()
  511. {
  512.     return this.instances[this.instances.length - 1].sourceText;
  513. }
  514.  
  515. ScriptManager.prototype.__defineGetter__ ("lineMap", smgr_linemap);
  516. function smgr_linemap()
  517. {
  518.     return this.instances[this.instances.length - 1].lineMap;
  519. }
  520.  
  521. ScriptManager.prototype.getInstanceBySequence =
  522. function smgr_bysequence (seq)
  523. {
  524.     for (var i = 0; i < this.instances.length; ++i)
  525.     {
  526.         if (this.instances[i].sequence == seq)
  527.             return this.instances[i];
  528.     }
  529.     
  530.     return null;
  531. }
  532.  
  533. ScriptManager.prototype.isLineExecutable =
  534. function smgr_isexe (line)
  535. {
  536.     for (var i in this.instances)
  537.     {
  538.         if (this.instances[i].isLineExecutable(line))
  539.             return true;
  540.     }
  541.  
  542.     return false;
  543. }
  544.  
  545. ScriptManager.prototype.hasBreakpoint =
  546. function smgr_hasbp (line)
  547. {
  548.     for (var i in this.instances)
  549.     {
  550.         if (this.instances[i].hasBreakpoint(line))
  551.             return true;
  552.     }
  553.     
  554.     return false;
  555. }
  556.  
  557. ScriptManager.prototype.setBreakpoint =
  558. function smgr_break (line, parentBP, props)
  559. {
  560.     var found = false;
  561.     
  562.     for (var i in this.instances)
  563.         found |= this.instances[i].setBreakpoint(line, parentBP, props);
  564.  
  565.     return found;
  566. }
  567.  
  568. ScriptManager.prototype.clearBreakpoint =
  569. function smgr_break (line)
  570. {
  571.     var found = false;
  572.  
  573.     for (var i in this.instances)
  574.         found |= this.instances[i].clearBreakpoint(line);
  575.  
  576.     return found;
  577. }
  578.  
  579. ScriptManager.prototype.hasFutureBreakpoint =
  580. function smgr_hasbp (line)
  581. {
  582.     var key = this.url + "#" + line;
  583.     return (key in console.fbreaks);
  584. }
  585.  
  586. ScriptManager.prototype.getFutureBreakpoint =
  587. function smgr_getfbp (line)
  588. {
  589.     return getFutureBreakpoint (this.url, line);
  590. }
  591.  
  592. ScriptManager.prototype.noteFutureBreakpoint =
  593. function smgr_fbreak (line, state)
  594. {
  595.     for (var i in this.instances)
  596.     {
  597.         if (this.instances[i]._lineMapInited)
  598.         {
  599.             if (state)
  600.             {
  601.                 arrayOrFlag (this.instances[i]._lineMap, line - 1, LINE_FBREAK);
  602.             }
  603.             else
  604.             {
  605.                 arrayAndFlag (this.instances[i]._lineMap, line - 1,
  606.                               ~LINE_FBREAK);
  607.             }
  608.         }
  609.     }
  610. }
  611.  
  612. function ScriptInstance (manager)
  613. {
  614.     this.scriptManager = manager;
  615.     this.url = manager.url;
  616.     this.creationDate = new Date();
  617.     this.topLevel = null;
  618.     this.functions = new Object();
  619.     this.nestLevel = 0;
  620.     this.isSealed = false;
  621.     this.scriptCount = 0;
  622.     this.breakpointCount = 0;
  623.     this.disabledScripts = 0;
  624.     this._lineMap = new Array();
  625.     this._lineMapInited = false;
  626. }
  627.  
  628. ScriptInstance.prototype.scanForMetaComments =
  629. function si_scan (start)
  630. {
  631.     const CHUNK_SIZE = 500;
  632.     const CHUNK_DELAY = 100;
  633.     
  634.     var scriptInstance = this;
  635.     var sourceText = this.sourceText;
  636.     
  637.     function onSourceLoaded(result)
  638.     {
  639.         if (result == Components.results.NS_OK)
  640.             scriptInstance.scanForMetaComments();
  641.     };
  642.     
  643.     if (!sourceText.isLoaded)
  644.     {
  645.         sourceText.loadSource(onSourceLoaded);
  646.         return;
  647.     }
  648.  
  649.     if (typeof start == "undefined")
  650.         start = 0;
  651.     
  652.     var end = Math.min (sourceText.lines.length, start + CHUNK_SIZE);
  653.     var obj = new Object();
  654.     
  655.     for (var i = start; i < end; ++i)
  656.     {
  657.         var ary = sourceText.lines[i].match (/\/\/@(\S+)(.*)/);
  658.         if (ary && ary[1] in console.metaDirectives && !(ary[1] in obj))
  659.         {
  660.             try
  661.             {
  662.                 console.metaDirectives[ary[1]](scriptInstance, i + 1, ary);
  663.             }
  664.             catch (ex)
  665.             {
  666.                 display (getMsg(MSN_ERR_META_FAILED, [ary[1], this.url, i + 1]),
  667.                          MT_ERROR);
  668.                 display (formatException (ex), MT_ERROR);
  669.             }
  670.         }
  671.     }
  672.  
  673.     if (i != sourceText.lines.length)
  674.         setTimeout (this.scanForMetaComments, CHUNK_DELAY, i);
  675. }
  676.  
  677. ScriptInstance.prototype.seal =
  678. function si_seal ()
  679. {
  680.     this.sealDate = new Date();
  681.     this.isSealed = true;
  682.  
  683.     if (isURLFiltered(this.url))
  684.     {
  685.         this.disabledScripts = 1;
  686.         var nada = SCRIPT_NODEBUG | SCRIPT_NOPROFILE;
  687.         if (this.topLevel && this.topLevel.isValid)
  688.             this.topLevel.jsdScript.flags |= nada;
  689.  
  690.         for (var f in this.functions)
  691.         {
  692.             if (this.functions[f].jsdScript.isValid)
  693.                 this.functions[f].jsdScript.flags |= nada;
  694.             ++this.disabledScripts;
  695.         }
  696.     }
  697.  
  698.     dispatch ("hook-script-instance-sealed", { scriptInstance: this });
  699. }
  700.  
  701. ScriptInstance.prototype.onScriptCreated =
  702. function si_created (scriptWrapper)
  703. {
  704.     var tag = scriptWrapper.jsdScript.tag;
  705.     
  706.     if (scriptWrapper.functionName)
  707.     {
  708.         this.functions[tag] = scriptWrapper;
  709.     }
  710.     else
  711.     {
  712.         this.topLevel = scriptWrapper;
  713.         scriptWrapper.functionName = MSG_VAL_TLSCRIPT;
  714.         scriptWrapper.addToLineMap(this._lineMap);
  715.         //var dummy = scriptWrapper.sourceText;
  716.         this.seal();
  717.     }
  718.  
  719.     ++this.scriptCount;
  720. }
  721.  
  722. ScriptInstance.prototype.onScriptInvalidated =
  723. function si_invalidated (scriptWrapper)
  724. {
  725.     //dd ("script invalidated");
  726.     scriptWrapper.clearBreakpoints();
  727.     --this.scriptCount;
  728. }
  729.  
  730. ScriptInstance.prototype.__defineGetter__ ("sourceText", si_gettext);
  731. function si_gettext ()
  732. {
  733.     if (!("_sourceText" in this))
  734.         this._sourceText = new SourceText (this);
  735.  
  736.     return this._sourceText;
  737. }
  738.  
  739. ScriptInstance.prototype.__defineGetter__ ("lineMap", si_linemap);
  740. function si_linemap()
  741. {
  742.     if (!this._lineMapInited)
  743.     {
  744.         if (this.topLevel && this.topLevel.jsdScript.isValid)
  745.             this.topLevel.addToLineMap(this._lineMap);
  746.         
  747.         for (var i in this.functions)
  748.         {
  749.             if (this.functions[i].jsdScript.isValid)
  750.                 this.functions[i].addToLineMap(this._lineMap);
  751.         }
  752.         
  753.         for (var fbp in console.fbreaks)
  754.         {
  755.             var fbreak = console.fbreaks[fbp];
  756.             if (fbreak.url == this.url)
  757.                 arrayOrFlag (this._lineMap, fbreak.lineNumber - 1, LINE_FBREAK);
  758.         }
  759.         
  760.         this._lineMapInited = true;
  761.     }
  762.  
  763.     return this._lineMap;            
  764. }
  765.  
  766. ScriptInstance.prototype.isLineExecutable =
  767. function si_isexe (line)
  768. {
  769.     if (this.topLevel && this.topLevel.jsdScript.isValid &&
  770.         this.topLevel.jsdScript.isLineExecutable (line, PCMAP_SOURCETEXT))
  771.     {
  772.         return true;
  773.     }
  774.     
  775.     for (var f in this.functions)
  776.     {
  777.         var jsdScript = this.functions[f].jsdScript;
  778.         if (line >= jsdScript.baseLineNumber &&
  779.             line <= jsdScript.baseLineNumber + jsdScript.lineExtent &&
  780.             jsdScript.isLineExecutable (line, PCMAP_SOURCETEXT))
  781.         {
  782.             return true;
  783.         }
  784.     }
  785.  
  786.     return false;
  787. }
  788.  
  789. ScriptInstance.prototype.hasBreakpoint =
  790. function si_hasbp (line)
  791. {
  792.     return Boolean (this.getBreakpoint(line));
  793. }
  794.  
  795. ScriptInstance.prototype.getBreakpoint =
  796. function si_getbp (line)
  797. {
  798.     for (var b in console.breaks)
  799.     {
  800.         if (console.breaks[b].scriptWrapper.scriptInstance == this)
  801.         {
  802.             if (typeof line == "undefined")
  803.                 return true;
  804.             
  805.             var jsdScript = console.breaks[b].scriptWrapper.jsdScript;
  806.             if (jsdScript.pcToLine(console.breaks[b].pc, PCMAP_SOURCETEXT) ==
  807.                 line)
  808.             {
  809.                 return console.breaks[b];
  810.             }
  811.         }
  812.     }
  813.  
  814.     return false;
  815. }
  816.  
  817. ScriptInstance.prototype.setBreakpoint =
  818. function si_setbp (line, parentBP, props)
  819. {
  820.     function setBP (scriptWrapper)
  821.     {
  822.         if (!scriptWrapper.jsdScript.isValid)
  823.             return false;
  824.  
  825.         var jsdScript = scriptWrapper.jsdScript;
  826.  
  827.         if (line >= jsdScript.baseLineNumber &&
  828.             line <= jsdScript.baseLineNumber + jsdScript.lineExtent &&
  829.             (jsdScript.isLineExecutable (line, PCMAP_SOURCETEXT) ||
  830.              jsdScript.baseLineNumber == line))
  831.         {
  832.             var pc = jsdScript.lineToPc(line, PCMAP_SOURCETEXT);
  833.             scriptWrapper.setBreakpoint(pc, parentBP, props);
  834.             return true;
  835.         }
  836.         return false;
  837.     };
  838.  
  839.     var found;
  840.     
  841.     if (this.topLevel)
  842.         found = setBP (this.topLevel);
  843.     for (var f in this.functions)
  844.         found |= setBP (this.functions[f]);
  845.  
  846.     if (this._lineMapInited && found)
  847.         arrayOrFlag(this._lineMap, line - 1, LINE_BREAK);
  848.  
  849.     return found;
  850. }
  851.  
  852. ScriptInstance.prototype.clearBreakpoint =
  853. function si_setbp (line)
  854. {
  855.     var found = false;
  856.     
  857.     function clearBP (scriptWrapper)
  858.     {
  859.         var jsdScript = scriptWrapper.jsdScript;
  860.         if (!jsdScript.isValid)
  861.             return;
  862.         
  863.         var pc = jsdScript.lineToPc(line, PCMAP_SOURCETEXT);
  864.         if (line >= jsdScript.baseLineNumber &&
  865.             line <= jsdScript.baseLineNumber + jsdScript.lineExtent &&
  866.             scriptWrapper.hasBreakpoint(pc))
  867.         {
  868.             found |= scriptWrapper.clearBreakpoint(pc);
  869.         }
  870.     };
  871.  
  872.     if (this._lineMapInited)
  873.         arrayAndFlag(this._lineMap, line - 1, ~LINE_BREAK);    
  874.     
  875.     if (this.topLevel)
  876.         clearBP (this.topLevel);
  877.  
  878.     for (var f in this.functions)
  879.         clearBP (this.functions[f]);
  880.  
  881.     return found;
  882. }
  883.  
  884. ScriptInstance.prototype.getScriptWrapperAtLine =
  885. function si_getscript (line)
  886. {
  887.     var targetScript = null;
  888.     var scriptWrapper;
  889.     
  890.     if (this.topLevel)
  891.     {
  892.         scriptWrapper = this.topLevel;
  893.         if (line >= scriptWrapper.jsdScript.baseLineNumber &&
  894.             line <= scriptWrapper.jsdScript.baseLineNumber +
  895.             scriptWrapper.jsdScript.lineExtent)
  896.         {
  897.             targetScript = scriptWrapper;
  898.         }
  899.     }    
  900.  
  901.     for (var f in this.functions)
  902.     {
  903.         scriptWrapper = this.functions[f];
  904.         if ((line >= scriptWrapper.jsdScript.baseLineNumber &&
  905.              line <= scriptWrapper.jsdScript.baseLineNumber +
  906.              scriptWrapper.jsdScript.lineExtent) &&
  907.             (!targetScript ||
  908.              scriptWrapper.jsdScript.lineExtent <
  909.              targetScript.jsdScript.lineExtent))
  910.         {
  911.             targetScript = scriptWrapper;
  912.         }
  913.     }
  914.  
  915.     return targetScript;
  916. }
  917.  
  918. ScriptInstance.prototype.containsScriptTag =
  919. function si_contains (tag)
  920. {
  921.     return ((this.topLevel && this.topLevel.tag == tag) ||
  922.             (tag in this.functions));
  923. }
  924.  
  925. ScriptInstance.prototype.guessFunctionNames =
  926. function si_guessnames ()
  927. {
  928.     var sourceLines = this._sourceText.lines;
  929.     var context = console.prefs["guessContext"];
  930.     var pattern = new RegExp (console.prefs["guessPattern"]);
  931.     var scanText;
  932.     
  933.     function getSourceContext (end)
  934.     {
  935.         var startLine = end - context;
  936.         if (startLine < 0)
  937.             startLine = 0;
  938.  
  939.         var text = "";
  940.         
  941.         for (i = startLine; i <= targetLine; ++i)
  942.             text += String(sourceLines[i]);
  943.     
  944.         var pos = text.lastIndexOf ("function");
  945.         if (pos != -1)
  946.             text = text.substring(0, pos);
  947.  
  948.         return text;
  949.     };
  950.         
  951.     for (var i in this.functions)
  952.     {
  953.         var scriptWrapper = this.functions[i];
  954.         if (scriptWrapper.jsdScript.functionName != "anonymous")
  955.             continue;
  956.         
  957.         var targetLine = scriptWrapper.jsdScript.baseLineNumber;
  958.         if (targetLine > sourceLines.length)
  959.         {
  960.             dd ("not enough source to guess function at line " + targetLine);
  961.             return;
  962.         }
  963.  
  964.         scanText = getSourceContext(targetLine);
  965.         var ary = scanText.match (pattern);
  966.         if (ary)
  967.         {
  968.             if ("charset" in this._sourceText)
  969.                 ary[1] = toUnicode(ary[1], this._sourceText.charset);
  970.             
  971.             scriptWrapper.functionName = getMsg(MSN_FMT_GUESSEDNAME, ary[1]);
  972.         }
  973.         else
  974.         {
  975.             if ("guessFallback" in console)
  976.             {
  977.                 var name = console.guessFallback(scriptWrapper, scanText);
  978.                 if (name)
  979.                 {
  980.                     scriptWrapper.functionName = getMsg(MSN_FMT_GUESSEDNAME,
  981.                                                         name);
  982.                 }
  983.             }
  984.         }
  985.     }
  986.  
  987.     dispatch ("hook-guess-complete", { scriptInstance: this });
  988. }
  989.  
  990. function ScriptWrapper (jsdScript)
  991. {    
  992.     this.jsdScript = jsdScript;
  993.     this.tag = jsdScript.tag;
  994.     this.functionName = jsdScript.functionName;
  995.     this.breakpointCount = 0;
  996.     this._lineMap = null;
  997.     this.breaks = new Object();
  998. }
  999.  
  1000. ScriptWrapper.prototype.__defineGetter__ ("sourceText", sw_getsource);
  1001. function sw_getsource ()
  1002. {
  1003.     if (!("_sourceText" in this))
  1004.     {
  1005.         if (!this.jsdScript.isValid)
  1006.             return null;
  1007.         this._sourceText = new PPSourceText(this);
  1008.     }
  1009.     
  1010.     return this._sourceText;
  1011. }
  1012.  
  1013. ScriptWrapper.prototype.__defineGetter__ ("lineMap", sw_linemap);
  1014. function sw_linemap ()
  1015. {
  1016.     if (!this._lineMap)
  1017.         this.addToLineMap(this._lineMap);
  1018.     
  1019.     return this._lineMap;
  1020. }
  1021.  
  1022. ScriptWrapper.prototype.hasBreakpoint =
  1023. function sw_hasbp (pc)
  1024. {
  1025.     var key = this.jsdScript.tag + ":" + pc;
  1026.     return key in console.breaks;
  1027. }
  1028.  
  1029. ScriptWrapper.prototype.getBreakpoint =
  1030. function sw_hasbp (pc)
  1031. {
  1032.     var key = this.jsdScript.tag + ":" + pc;
  1033.     if (key in console.breaks)
  1034.         return console.breaks[key];
  1035.     
  1036.     return null;
  1037. }
  1038.  
  1039. ScriptWrapper.prototype.setBreakpoint =
  1040. function sw_setbp (pc, parentBP, props)
  1041. {
  1042.     var key = this.jsdScript.tag + ":" + pc;
  1043.     
  1044.     //dd ("setting breakpoint in " + this.functionName + " " + key);
  1045.     
  1046.     if (key in console.breaks)
  1047.         return null;
  1048.  
  1049.     var brk = new BreakInstance (parentBP, this, pc);
  1050.     if (props)
  1051.         brk.setProperties(props);
  1052.     
  1053.     console.breaks[key] = brk;
  1054.     this.breaks[key] = brk;
  1055.     
  1056.     if (parentBP)
  1057.     {
  1058.         parentBP.childrenBP[key] = brk;
  1059.         brk.lineNumber = parentBP.lineNumber;
  1060.         brk.url = parentBP.url;
  1061.     }
  1062.  
  1063.     if ("_sourceText" in this)
  1064.     {
  1065.         var line = this.jsdScript.pcToLine(brk.pc, PCMAP_PRETTYPRINT);
  1066.         arrayOrFlag (this._sourceText.lineMap, line - 1, LINE_BREAK);
  1067.     }
  1068.  
  1069.     ++this.scriptInstance.breakpointCount;
  1070.     ++this.breakpointCount;
  1071.     
  1072.     if (this.scriptInstance._lineMapInited)
  1073.     {
  1074.         line = this.jsdScript.pcToLine (pc, PCMAP_SOURCETEXT);
  1075.         arrayOrFlag (this.scriptInstance._lineMap, line - 1, LINE_BREAK);
  1076.     }
  1077.  
  1078.     dispatch ("hook-break-set", { breakWrapper: brk });
  1079.     
  1080.     return brk;
  1081. }
  1082.  
  1083. ScriptWrapper.prototype.clearBreakpoints =
  1084. function sw_clearbps ()
  1085. {
  1086.     var found = false;
  1087.     
  1088.     for (b in this.breaks)
  1089.         found |= this.clearBreakpoint(this.breaks[b].pc);
  1090.  
  1091.     return found;
  1092. }
  1093.  
  1094. ScriptWrapper.prototype.clearBreakpoint =
  1095. function sw_clearbp (pc)
  1096. {
  1097.     var key = this.jsdScript.tag + ":" + pc;
  1098.     if (!(key in console.breaks))
  1099.         return false;
  1100.  
  1101.     var brk = console.breaks[key];
  1102.  
  1103.     if ("propsWindow" in brk)
  1104.         brk.propsWindow.close();
  1105.     
  1106.     delete console.breaks[key];
  1107.     delete this.breaks[key];
  1108.  
  1109.     if (brk.parentBP)
  1110.         delete brk.parentBP.childrenBP[key];
  1111.  
  1112.     var line;
  1113.     
  1114.     if ("_sourceText" in this && this.jsdScript.isValid)
  1115.     {
  1116.         line = this.jsdScript.pcToLine(brk.pc, PCMAP_PRETTYPRINT);
  1117.         this._sourceText.lineMap[line - 1] &= ~LINE_BREAK;
  1118.     }
  1119.     
  1120.     --this.scriptInstance.breakpointCount;
  1121.     --this.breakpointCount;
  1122.  
  1123.     if (this.scriptInstance._lineMapInited)
  1124.     {
  1125.         if (this.jsdScript.isValid)
  1126.         {
  1127.             line = this.jsdScript.pcToLine (pc, PCMAP_SOURCETEXT);
  1128.             if (!this.scriptInstance.hasBreakpoint(line))
  1129.                 this.scriptInstance._lineMap[line - 1] &= ~LINE_BREAK;
  1130.         }
  1131.         else
  1132.         {
  1133.             /* script is gone, no way to find out where the break actually
  1134.              * was, so we have to redo the whole map. */
  1135.             this.scriptInstance._lineMapInited = false;
  1136.             this.scriptInstance._lineMap.length = 0;
  1137.             var dummy = this.scriptInstance.lineMap;
  1138.         }
  1139.     }
  1140.  
  1141.     dispatch ("hook-break-clear", { breakWrapper: brk });
  1142.  
  1143.     if (this.jsdScript.isValid)
  1144.         this.jsdScript.clearBreakpoint (pc);
  1145.  
  1146.     return true;
  1147. }
  1148.  
  1149. ScriptWrapper.prototype.addToLineMap =
  1150. function sw_addmap (lineMap)
  1151. {    
  1152.     var jsdScript = this.jsdScript;
  1153.     var end = jsdScript.baseLineNumber + jsdScript.lineExtent;
  1154.     for (var i = jsdScript.baseLineNumber; i < end; ++i)
  1155.     {
  1156.         if (jsdScript.isLineExecutable(i, PCMAP_SOURCETEXT))
  1157.             arrayOrFlag (lineMap, i - 1, LINE_BREAKABLE);
  1158.     }
  1159.  
  1160.     for (i in this.breaks)
  1161.     {
  1162.         var line = jsdScript.pcToLine(this.breaks[i].pc, PCMAP_SOURCETEXT);
  1163.         arrayOrFlag (lineMap, line - 1, LINE_BREAK);
  1164.     }
  1165. }
  1166.  
  1167. function getScriptWrapper(jsdScript)
  1168. {
  1169.     if (!ASSERT(jsdScript, "getScriptWrapper: null jsdScript"))
  1170.         return null;
  1171.     
  1172.     var tag = jsdScript.tag;
  1173.     if (tag in console.scriptWrappers)
  1174.         return console.scriptWrappers[tag];
  1175.  
  1176.     dd ("Can't find a wrapper for " + formatScript(jsdScript));
  1177.     return null;
  1178. }
  1179.  
  1180. function BreakInstance (parentBP, scriptWrapper, pc)
  1181. {
  1182.     this._enabled = true;
  1183.     this.parentBP = parentBP;
  1184.     this.scriptWrapper = scriptWrapper;
  1185.     this.pc = pc;
  1186.     this.url = scriptWrapper.jsdScript.fileName;
  1187.     this.lineNumber = scriptWrapper.jsdScript.pcToLine (pc, PCMAP_SOURCETEXT);
  1188.     this.oneTime = false;
  1189.     this.triggerCount = 0;
  1190.     
  1191.     scriptWrapper.jsdScript.setBreakpoint (pc);
  1192. }
  1193.  
  1194. BreakInstance.prototype.__defineGetter__ ("jsdURL", bi_getURL);
  1195. function bi_getURL ()
  1196. {
  1197.     return ("x-jsd:break?url=" + encodeURIComponent(this.url) +
  1198.             "&lineNumber=" + this.lineNumber +
  1199.             "&conditionEnabled=" + this.conditionEnabled +
  1200.             "&condition=" + encodeURIComponent(this.condition) +
  1201.             "&passExceptions=" + this.passExceptions +
  1202.             "&logResult=" + this.logResult +
  1203.             "&resultAction=" + this.resultAction +
  1204.             "&enabled=" + this.enabled);
  1205. }
  1206.  
  1207. BreakInstance.prototype.getProperties =
  1208. function bi_getprops()
  1209. {
  1210.     var rv = new Object();
  1211.  
  1212.     rv.enabled = this._enabled;
  1213.     if ("_conditionEnabled" in this)
  1214.         rv.conditionEnabled = this._conditionEnabled;
  1215.     if ("_condition" in this)
  1216.         rv.condition = this._condition;
  1217.     if ("_passExceptions" in this)
  1218.         rv.passExceptions = this._passExceptions;
  1219.     if ("_logResult" in this)
  1220.         rv.logResult = this._logResult;
  1221.     if ("_resultAction" in this)
  1222.         rv.resultAction = this._resultAction;
  1223.  
  1224.     return rv;
  1225. }
  1226.  
  1227. BreakInstance.prototype.setProperties =
  1228. function bi_setprops(obj)
  1229. {
  1230.     for (var p in obj)
  1231.     {
  1232.         if (p.search(/pc|url|lineNumber/) == -1)
  1233.             this[p] = obj[p];
  1234.     }
  1235.     
  1236.     if ("propsWindow" in this)
  1237.         this.propsWindow.populateFromBreakpoint();
  1238. }
  1239.  
  1240. BreakInstance.prototype.clearBreakpoint =
  1241. function bi_clear()
  1242. {
  1243.     this.scriptWrapper.clearBreakpoint(this.pc);
  1244. }
  1245.  
  1246. BreakInstance.prototype.__defineGetter__ ("enabled", bi_getEnabled);
  1247. function bi_getEnabled ()
  1248. {
  1249.     return this._enabled;
  1250. }
  1251.  
  1252. BreakInstance.prototype.__defineSetter__ ("enabled", bi_setEnabled);
  1253. function bi_setEnabled (state)
  1254. {
  1255.     if (state != this._enabled)
  1256.     {
  1257.         this._enabled = state;
  1258.         if (state)
  1259.             this.scriptWrapper.jsdScript.setBreakpoint(this.pc);
  1260.         else
  1261.             this.scriptWrapper.jsdScript.clearBreakpoint(this.pc);
  1262.     }
  1263.     
  1264.     return state;
  1265. }
  1266.  
  1267. BreakInstance.prototype.__defineGetter__ ("conditionEnabled", bi_getCondEnabled);
  1268. function bi_getCondEnabled ()
  1269. {
  1270.     if ("_conditionEnabled" in this)
  1271.         return this._conditionEnabled;
  1272.     
  1273.     if (this.parentBP)
  1274.         return this.parentBP.conditionEnabled;
  1275.  
  1276.     return false;
  1277. }
  1278.  
  1279. BreakInstance.prototype.__defineSetter__ ("conditionEnabled", bi_setCondEnabled);
  1280. function bi_setCondEnabled (state)
  1281. {
  1282.     if (this.parentBP)
  1283.         return this.parentBP.conditionEnabled = state;
  1284.     
  1285.     return this._conditionEnabled = state;
  1286. }
  1287.  
  1288. BreakInstance.prototype.__defineGetter__ ("condition", bi_getCondition);
  1289. function bi_getCondition ()
  1290. {
  1291.     if ("_condition" in this)
  1292.         return this._condition;
  1293.     
  1294.     if (this.parentBP)
  1295.         return this.parentBP.condition;
  1296.  
  1297.     return "";
  1298. }
  1299.  
  1300. BreakInstance.prototype.__defineSetter__ ("condition", bi_setCondition);
  1301. function bi_setCondition (value)
  1302. {
  1303.     if (this.parentBP)
  1304.         return this.parentBP.condition = value;
  1305.     
  1306.     return this._condition = value;
  1307. }
  1308.  
  1309.  
  1310. BreakInstance.prototype.__defineGetter__ ("passExceptions", bi_getException);
  1311. function bi_getException ()
  1312. {
  1313.     if ("_passExceptions" in this)
  1314.         return this._passExceptions;
  1315.     
  1316.     if (this.parentBP)
  1317.         return this.parentBP.passExceptions;
  1318.  
  1319.     return false;
  1320. }
  1321.  
  1322. BreakInstance.prototype.__defineSetter__ ("passExceptions", bi_setException);
  1323. function bi_setException (state)
  1324. {
  1325.     if (this.parentBP)
  1326.         return this.parentBP.passExceptions = state;
  1327.     
  1328.     return this._passExceptions = state;
  1329. }
  1330.  
  1331.  
  1332. BreakInstance.prototype.__defineGetter__ ("logResult", bi_getLogResult);
  1333. function bi_getLogResult ()
  1334. {
  1335.     if ("_logResult" in this)
  1336.         return this._logResult;
  1337.     
  1338.     if (this.parentBP)
  1339.         return this.parentBP.logResult;
  1340.  
  1341.     return false;
  1342. }
  1343.  
  1344. BreakInstance.prototype.__defineSetter__ ("logResult", bi_setLogResult);
  1345. function bi_setLogResult (state)
  1346. {
  1347.     if (this.parentBP)
  1348.         return this.parentBP.logResult = state;
  1349.     
  1350.     return this._logResult = state;
  1351. }
  1352.  
  1353.  
  1354. BreakInstance.prototype.__defineGetter__ ("resultAction", bi_getResultAction);
  1355. function bi_getResultAction ()
  1356. {
  1357.     if ("_resultAction" in this)
  1358.         return this._resultAction;
  1359.     
  1360.     if (this.parentBP)
  1361.         return this.parentBP.resultAction;
  1362.  
  1363.     return BREAKPOINT_STOPALWAYS;
  1364. }
  1365.  
  1366. BreakInstance.prototype.__defineSetter__ ("resultAction", bi_setResultAction);
  1367. function bi_setResultAction (state)
  1368. {
  1369.     if (this.parentBP)
  1370.         return this.parentBP.resultAction = state;
  1371.     
  1372.     return this._resultAction = state;
  1373. }
  1374.  
  1375. function FutureBreakpoint (url, lineNumber)
  1376. {
  1377.     this.url = url;
  1378.     this.lineNumber = lineNumber;
  1379.     this.enabled = true;
  1380.     this.childrenBP = new Object();
  1381.     this.conditionEnabled = false;
  1382.     this.condition = "";
  1383.     this.passExceptions = false;
  1384.     this.logResult = false;
  1385.     this.resultAction = BREAKPOINT_STOPALWAYS;
  1386. }
  1387.  
  1388. FutureBreakpoint.prototype.__defineGetter__ ("jsdURL", fb_getURL);
  1389. function fb_getURL ()
  1390. {
  1391.     return ("x-jsd:fbreak?url=" + encodeURIComponent(this.url) +
  1392.             "&lineNumber=" + this.lineNumber +
  1393.             "&conditionEnabled=" + this.conditionEnabled +
  1394.             "&condition=" + encodeURIComponent(this.condition) +
  1395.             "&passExceptions=" + this.passExceptions +
  1396.             "&logResult=" + this.logResult +
  1397.             "&resultAction=" + this.resultAction +
  1398.             "&enabled=" + this.enabled);
  1399. }
  1400.  
  1401.  
  1402. FutureBreakpoint.prototype.getProperties =
  1403. function fb_getprops()
  1404. {
  1405.     var rv = new Object();
  1406.     
  1407.     rv.conditionEnabled = this.conditionEnabled;
  1408.     rv.condition = this.condition;
  1409.     rv.passExceptions = this.passExceptions;
  1410.     rv.logResult = this.logResult;
  1411.     rv.resultAction = this.resultAction;
  1412.  
  1413.     return rv;
  1414. }
  1415.  
  1416. FutureBreakpoint.prototype.setProperties =
  1417. function fb_setprops(obj)
  1418. {
  1419.     for (var p in obj)
  1420.     {
  1421.         if (p.search(/url|lineNumber|childrenBP/) == -1)
  1422.             this[p] = obj[p];
  1423.     }
  1424.     
  1425.     if ("propsWindow" in this)
  1426.         this.propsWindow.populateFromBreakpoint();
  1427. }
  1428.  
  1429. FutureBreakpoint.prototype.clearFutureBreakpoint =
  1430. function fb_clear ()
  1431. {
  1432.     clearFutureBreakpoint (this.url, this.lineNumber);
  1433. }
  1434.  
  1435. FutureBreakpoint.prototype.resetInstances =
  1436. function fb_reseti ()
  1437. {
  1438.     for (var url in console.scriptManagers)
  1439.     {
  1440.         if (url.indexOf(this.url) != -1)
  1441.             console.scriptManagers[url].setBreakpoint(this.lineNumber);
  1442.     }
  1443. }
  1444.  
  1445. FutureBreakpoint.prototype.clearInstances =
  1446. function fb_cleari ()
  1447. {
  1448.     for (var url in console.scriptManagers)
  1449.     {
  1450.         if (url.indexOf(this.url) != -1)
  1451.             console.scriptManagers[url].clearBreakpoint(this.lineNumber);
  1452.     }
  1453. }
  1454.  
  1455. function testBreakpoint(currentFrame, rv)
  1456. {
  1457.     var tag = currentFrame.script.tag;
  1458.     if (!(tag in console.scriptWrappers))
  1459.         return -1;
  1460.     
  1461.     var scriptWrapper = console.scriptWrappers[tag];
  1462.     var breakpoint = scriptWrapper.getBreakpoint(currentFrame.pc);
  1463.     if (!ASSERT(breakpoint, "can't find breakpoint for " +
  1464.                 formatFrame(currentFrame)))
  1465.     {
  1466.         return -1;
  1467.     }
  1468.  
  1469.     if (!ASSERT(breakpoint.enabled, "stopped at a disabled breakpoint?"))
  1470.         return RETURN_CONTINUE;
  1471.  
  1472.     ++breakpoint.triggerCount;
  1473.     if ("propsWindow" in breakpoint)
  1474.         breakpoint.propsWindow.onBreakpointTriggered();
  1475.     
  1476.     if (breakpoint.oneTime)
  1477.         scriptWrapper.clearBreakpoint(currentFrame.pc);
  1478.  
  1479.     if (breakpoint.conditionEnabled && breakpoint.condition)
  1480.     {
  1481.         var result = new Object();
  1482.         var script = "var __trigger__ = function (__count__) {" +
  1483.             breakpoint.condition + "}; __trigger__.apply(this, [" +
  1484.             breakpoint.triggerCount + "]);";
  1485.         if (!currentFrame.eval (script,
  1486.                                 JSD_URL_SCHEME + "breakpoint-condition",
  1487.                                 1, result))
  1488.         {
  1489.             /* condition raised an exception */
  1490.             if (breakpoint.passExceptions)
  1491.             {
  1492.                 rv.value = result.value;
  1493.                 return RETURN_THROW;
  1494.             }
  1495.                 
  1496.             display (MSG_ERR_CONDITION_FAILED, MT_ERROR);
  1497.             display (formatException(result.value.getWrappedValue()), MT_ERROR);
  1498.         }
  1499.         else
  1500.         {
  1501.             /* condition executed ok */
  1502.             if (breakpoint.logResult)
  1503.             {
  1504.                 display (result.value.stringValue, MT_LOG);
  1505.             }
  1506.  
  1507.             if (breakpoint.resultAction == BREAKPOINT_EARLYRETURN)
  1508.             {
  1509.                 rv.value = result.value;
  1510.                 return RETURN_VALUE;
  1511.             }
  1512.  
  1513.             if (breakpoint.resultAction == BREAKPOINT_STOPNEVER ||
  1514.                 (breakpoint.resultAction == BREAKPOINT_STOPTRUE &&
  1515.                  !result.value.booleanValue))
  1516.             {
  1517.                 return RETURN_CONTINUE;
  1518.             }            
  1519.         }
  1520.     }
  1521.  
  1522.     return -1;
  1523. }
  1524.  
  1525. const EMODE_IGNORE = 0;
  1526. const EMODE_TRACE  = 1;
  1527. const EMODE_BREAK  = 2;
  1528.  
  1529. const TMODE_IGNORE = 0;
  1530. const TMODE_TRACE  = 1;
  1531. const TMODE_BREAK  = 2;
  1532.  
  1533. function debugTrap (frames, type, rv)
  1534. {
  1535.     var tn = "";
  1536.     var retcode = jsdIExecutionHook.RETURN_CONTINUE;
  1537.  
  1538.     //dd ("debugTrap");
  1539.  
  1540.     var frame = frames[0];
  1541.     
  1542.     $ = new Array();
  1543.     
  1544.     switch (type)
  1545.     {
  1546.         case jsdIExecutionHook.TYPE_BREAKPOINT:
  1547.             var bpResult = testBreakpoint(frame, rv);
  1548.             if (bpResult != -1)
  1549.                 return bpResult;
  1550.             tn = MSG_VAL_BREAKPOINT;
  1551.             break;
  1552.         case jsdIExecutionHook.TYPE_DEBUG_REQUESTED:
  1553.             tn = MSG_VAL_DEBUG;
  1554.             break;
  1555.         case jsdIExecutionHook.TYPE_DEBUGGER_KEYWORD:
  1556.             tn = MSG_VAL_DEBUGGER;
  1557.             break;
  1558.         case jsdIExecutionHook.TYPE_THROW:
  1559.             dd (dumpObjectTree(rv));
  1560.             display (getMsg(MSN_EXCEPTION_TRACE,
  1561.                             [rv.value.stringValue, formatFrame(frame)]),
  1562.                      MT_ETRACE);
  1563.             if (rv.value.jsClassName == "Error")
  1564.                 display (formatProperty(rv.value.getProperty("message")),
  1565.                          MT_EVAL_OUT);
  1566.  
  1567.             if (console.throwMode != TMODE_BREAK)
  1568.                 return jsdIExecutionHook.RETURN_CONTINUE_THROW;
  1569.  
  1570.             console.currentException = rv.value;
  1571.             retcode = jsdIExecutionHook.RETURN_CONTINUE_THROW;
  1572.             
  1573.             tn = MSG_VAL_THROW;
  1574.             break;
  1575.         case jsdIExecutionHook.TYPE_INTERRUPTED:
  1576.             
  1577.             if (!frame.script.functionName && 
  1578.                 isURLFiltered(frame.script.fileName))
  1579.             {
  1580.                 //dd ("filtered url: " + frame.script.fileName);
  1581.                 frame.script.flags |= SCRIPT_NOPROFILE | SCRIPT_NODEBUG;
  1582.                 return retcode;
  1583.             }
  1584.  
  1585.             var line;
  1586.             if (console.prefs["prettyprint"])
  1587.                 line = frame.script.pcToLine (frame.pc, PCMAP_PRETTYPRINT);
  1588.             else
  1589.                 line = frame.line;
  1590.             if (console._stepPast == 
  1591.                 frames.length + frame.script.fileName + line)
  1592.             {
  1593.                 //dd("stepPast: " + console._stepPast);
  1594.                 return retcode;
  1595.             }
  1596.             
  1597.             delete console._stepPast;
  1598.             setStopState(false);
  1599.             break;
  1600.         default:
  1601.             /* don't print stop/cont messages for other types */
  1602.     }
  1603.  
  1604.     console.jsds.functionHook = null;
  1605.  
  1606.     /* set our default return value */
  1607.     console._continueCodeStack.push (retcode);
  1608.  
  1609.     if (tn)
  1610.         display (getMsg(MSN_STOP, tn), MT_STOP);
  1611.     
  1612.     /* build an array of frames */
  1613.     console.frames = frames;
  1614.     
  1615.     console.trapType = type;
  1616.     
  1617.     try
  1618.     {    
  1619.         console.jsds.enterNestedEventLoop({onNest: eventLoopNested}); 
  1620.     }
  1621.     catch (ex)
  1622.     {
  1623.         dd ("caught " + ex + " while nested");
  1624.     }
  1625.     
  1626.     /* execution pauses here until someone calls exitNestedEventLoop() */
  1627.  
  1628.     clearCurrentFrame();
  1629.     rv.value = ("currentException" in console) ? console.currentException : null;
  1630.  
  1631.     delete console.frames;
  1632.     delete console.trapType;
  1633.     delete console.currentException;
  1634.     $ = new Array();
  1635.     
  1636.     dispatch ("hook-debug-continue");
  1637.  
  1638.     if (tn)
  1639.         display (getMsg(MSN_CONT, tn), MT_CONT);
  1640.  
  1641.     return console._continueCodeStack.pop();
  1642. }
  1643.  
  1644. function eventLoopNested ()
  1645. {
  1646.     window.focus();
  1647.     window.getAttention();
  1648.  
  1649.     dispatch ("hook-debug-stop");
  1650. }
  1651.  
  1652. function getCurrentFrame()
  1653. {
  1654.     if ("frames" in console)
  1655.         return console.frames[console._currentFrameIndex];
  1656.  
  1657.     return null;
  1658. }
  1659.  
  1660. function getCurrentFrameIndex()
  1661. {
  1662.     if (typeof console._currentFrameIndex == "undefined")
  1663.         return -1;
  1664.     
  1665.     return console._currentFrameIndex;
  1666. }
  1667.  
  1668. function setCurrentFrameByIndex (index)
  1669. {
  1670.     if (!console.frames)
  1671.         throw new BadMojo (ERR_NO_STACK);
  1672.     
  1673.     ASSERT (index >= 0 && index < console.frames.length, "index out of range");
  1674.  
  1675.     console._currentFrameIndex = index;
  1676.     var cf = console.frames[console._currentFrameIndex];
  1677.     dispatch ("set-eval-obj", { jsdValue: cf });
  1678.     console.stopFile = (cf.isNative) ? MSG_URL_NATIVE : cf.script.fileName;
  1679.     console.stopLine = cf.line;
  1680.     delete console._pp_stopLine;
  1681.     return cf;
  1682. }
  1683.  
  1684. function clearCurrentFrame ()
  1685. {
  1686.     if (!console.frames)
  1687.         throw new BadMojo (ERR_NO_STACK);
  1688.  
  1689.     if (console.currentEvalObject instanceof jsdIStackFrame)
  1690.         dispatch ("set-eval-obj", { jsdValue: console.jsdConsole });
  1691.     
  1692.     delete console.stopLine;
  1693.     delete console._pp_stopLine;
  1694.     delete console.stopFile;
  1695.     delete console._currentFrameIndex;
  1696. }
  1697.  
  1698. function formatArguments (v)
  1699. {
  1700.     if (!v)
  1701.         return "";
  1702.  
  1703.     var ary = new Array();
  1704.     var p = new Object();
  1705.     v.getProperties (p, {});
  1706.     p = p.value;
  1707.     for (var i = 0; i < p.length; ++i)
  1708.     {
  1709.         if (p[i].flags & jsdIProperty.FLAG_ARGUMENT)
  1710.             ary.push (getMsg(MSN_FMT_ARGUMENT,
  1711.                              [p[i].name.stringValue,
  1712.                               formatValue(p[i].value, FTYPE_SUMMARY)]));
  1713.     }
  1714.     
  1715.     return ary.join (MSG_COMMASP); 
  1716. }
  1717.  
  1718. function formatFlags (flags)
  1719. {
  1720.     var s = "";
  1721.     
  1722.     if (flags & PROP_ENUMERATE)
  1723.         s += MSG_VF_ENUMERABLE;
  1724.     if (flags & PROP_READONLY)
  1725.         s += MSG_VF_READONLY;        
  1726.     if (flags & PROP_PERMANENT)
  1727.         s += MSG_VF_PERMANENT;
  1728.     if (flags & PROP_ALIAS)
  1729.         s += MSG_VF_ALIAS;
  1730.     if (flags & PROP_ARGUMENT)
  1731.         s += MSG_VF_ARGUMENT;
  1732.     if (flags & PROP_VARIABLE)
  1733.         s += MSG_VF_VARIABLE;
  1734.     if (flags & PROP_ERROR)
  1735.         s += MSG_VF_ERROR;
  1736.     if (flags & PROP_EXCEPTION)
  1737.         s += MSG_VF_EXCEPTION;
  1738.     if (flags & PROP_HINTED)
  1739.         s += MSG_VF_HINTED;
  1740.  
  1741.     return s;
  1742. }
  1743.  
  1744. function formatProperty (p, formatType)
  1745. {
  1746.     if (!p)
  1747.         throw new BadMojo (ERR_REQUIRED_PARAM, "p");
  1748.  
  1749.     var s = formatFlags (p.flags);
  1750.  
  1751.     if (formatType == FTYPE_ARRAY)
  1752.     {
  1753.         var rv = formatValue (p.value, FTYPE_ARRAY);
  1754.         return [p.name.stringValue, rv[1] ? rv[1] : rv[0], rv[2], s];
  1755.     }
  1756.     
  1757.     return getMsg(MSN_FMT_PROPERTY, [s, p.name.stringValue,
  1758.                                      formatValue(p.value)]);
  1759. }
  1760.  
  1761. function formatScript (script)
  1762. {
  1763.     if (!script)
  1764.         throw new BadMojo (ERR_REQUIRED_PARAM, "script");
  1765.  
  1766.     var functionName;
  1767.     if (script.tag in console.scriptWrappers)
  1768.         functionName = console.scriptWrappers[script.tag].functionName;
  1769.     else
  1770.         functionName = script.functionName;
  1771.     return getMsg (MSN_FMT_SCRIPT, [functionName, script.fileName]);
  1772. }
  1773.  
  1774. function formatFrame (f)
  1775. {
  1776.     if (!f)
  1777.         throw new BadMojo (ERR_REQUIRED_PARAM, "f");
  1778.     var url = (f.isNative) ? MSG_URL_NATIVE : f.script.fileName;
  1779.     return getMsg (MSN_FMT_FRAME,
  1780.                    [f.functionName, formatArguments(f.scope), url, f.line]);
  1781. }
  1782.  
  1783. function formatValue (v, formatType)
  1784. {
  1785.     if (!v)
  1786.         throw new BadMojo (ERR_REQUIRED_PARAM, "v");
  1787.  
  1788.     if (!(v instanceof jsdIValue))
  1789.         throw new BadMojo (ERR_INVALID_PARAM, "v", String(v));
  1790.  
  1791.     var type;
  1792.     var value;
  1793.         
  1794.     switch (v.jsType)
  1795.     {
  1796.         case jsdIValue.TYPE_BOOLEAN:
  1797.             type = MSG_TYPE_BOOLEAN;
  1798.             value = String(v.booleanValue);
  1799.             break;
  1800.         case jsdIValue.TYPE_DOUBLE:
  1801.             type = MSG_TYPE_DOUBLE;
  1802.             value = v.doubleValue;
  1803.             break;
  1804.         case jsdIValue.TYPE_INT:
  1805.             type = MSG_TYPE_INT;
  1806.             value = v.intValue;
  1807.             break;
  1808.         case jsdIValue.TYPE_FUNCTION:
  1809.             type = MSG_TYPE_FUNCTION;
  1810.             value = v.jsFunctionName;
  1811.             break;
  1812.         case jsdIValue.TYPE_NULL:
  1813.             type = MSG_TYPE_NULL;
  1814.             value = MSG_TYPE_NULL;
  1815.             break;
  1816.         case jsdIValue.TYPE_OBJECT:
  1817.             if (formatType == FTYPE_STD)
  1818.             {
  1819.                 type = MSG_TYPE_OBJECT;
  1820.                 value = getMsg(MSN_FMT_OBJECT, String(v.propertyCount));
  1821.             }
  1822.             else
  1823.             {
  1824.                 if (v.jsClassName)
  1825.                     if (v.jsClassName == "XPCWrappedNative_NoHelper")
  1826.                         type = MSG_CLASS_XPCOBJ;
  1827.                     else
  1828.                         type = v.jsClassName;
  1829.                 else
  1830.                     type = MSG_TYPE_OBJECT;
  1831.                 value = "{" + String(v.propertyCount) + "}";
  1832.             }
  1833.             break;
  1834.         case jsdIValue.TYPE_STRING:
  1835.             type = MSG_TYPE_STRING;
  1836.             var strval = v.stringValue;
  1837.             if (strval.length > console.prefs["maxStringLength"])
  1838.                 strval = getMsg(MSN_FMT_LONGSTR, strval.length);
  1839.             else
  1840.                 strval = strval.quote()
  1841.             value = strval;
  1842.             break;
  1843.         case jsdIValue.TYPE_VOID:
  1844.             type = MSG_TYPE_VOID;
  1845.             value = MSG_TYPE_VOID;            
  1846.             break;
  1847.         default:
  1848.             type = MSG_TYPE_UNKNOWN;
  1849.             value = MSG_TYPE_UNKNOWN;
  1850.             break;
  1851.     }
  1852.  
  1853.     if (formatType == FTYPE_SUMMARY)
  1854.         return getMsg (MSN_FMT_VALUE_SHORT, [type, value]);
  1855.  
  1856.     var className;
  1857.     if (v.jsClassName)
  1858.         if (v.jsClassName == "XPCWrappedNative_NoHelper")
  1859.             /* translate this long, unintuitive, and common class name into
  1860.              * something more palatable. */
  1861.             className = MSG_CLASS_XPCOBJ;
  1862.         else
  1863.             className = v.jsClassName;
  1864.  
  1865.     if (formatType == FTYPE_ARRAY)
  1866.         return [type, className, value];
  1867.  
  1868.     if (className)
  1869.         return getMsg (MSN_FMT_VALUE_LONG, [type, v.jsClassName, value]);
  1870.  
  1871.     return getMsg (MSN_FMT_VALUE_MED, [type, value]);
  1872.  
  1873. }
  1874.  
  1875. function displayCallStack ()
  1876. {
  1877.     for (var i = 0; i < console.frames.length; ++i)
  1878.         displayFrame (console.frames[i], i);
  1879. }
  1880.  
  1881. function displayProperties (v)
  1882. {
  1883.     if (!v)
  1884.         throw new BadMojo (ERR_REQUIRED_PARAM, "v");
  1885.  
  1886.     if (!(v instanceof jsdIValue))
  1887.         throw new BadMojo (ERR_INVALID_PARAM, "v", String(v));
  1888.  
  1889.     var p = new Object();
  1890.     v.getProperties (p, {});
  1891.     for (var i in p.value) display(formatProperty (p.value[i]), MT_EVAL_OUT);
  1892. }
  1893.  
  1894. function displaySourceContext (sourceText, line, contextLines)
  1895. {
  1896.     function onSourceLoaded (status)
  1897.     {
  1898.         if (status == Components.results.NS_OK)
  1899.             displaySourceContext (sourceText, line, contextLines);
  1900.     }
  1901.     
  1902.     if (sourceText.isLoaded)
  1903.     {
  1904.         for (var i = line - contextLines; i <= line + contextLines; ++i)
  1905.         {
  1906.             if (i > 0 && i < sourceText.lines.length)
  1907.             {
  1908.                 var sourceLine;
  1909.                 if ("charset" in sourceText)
  1910.                 {
  1911.                     sourceLine = toUnicode(sourceText.lines[i - 1],
  1912.                                      sourceText.charset);
  1913.                 }
  1914.                 else
  1915.                 {
  1916.                     sourceLine = sourceText.lines[i - 1];
  1917.                 }
  1918.                 
  1919.                 display (getMsg(MSN_SOURCE_LINE, [zeroPad (i, 3), sourceLine]),
  1920.                          i == line ? MT_STEP : MT_SOURCE);
  1921.             }
  1922.         }
  1923.     }
  1924.     else
  1925.     {
  1926.         sourceText.loadSource (onSourceLoaded);
  1927.     }
  1928. }
  1929.     
  1930. function displayFrame (jsdFrame, idx, showHeader, sourceContext)
  1931. {
  1932.     if (typeof idx == "undefined")
  1933.     {
  1934.         for (idx = 0; idx < console.frames.length; ++idx)
  1935.             if (jsdFrame == console.frames[idx])
  1936.                 break;
  1937.     
  1938.         if (idx >= console.frames.length)
  1939.             idx = MSG_VAL_UNKNOWN;
  1940.     }
  1941.  
  1942.     if (typeof showHeader == "undefined")
  1943.         showHeader = true;
  1944.  
  1945.     if (typeof sourceContext == "undefined")
  1946.         sourceContext = null;
  1947.  
  1948.     display(getMsg(MSN_FMT_FRAME_LINE, [idx, formatFrame(jsdFrame)]), MT_OUTPUT);
  1949.  
  1950.     if (!jsdFrame.isNative && sourceContext != null)
  1951.     {
  1952.         var jsdScript = jsdFrame.script;
  1953.         var scriptWrapper = getScriptWrapper(jsdScript);
  1954.  
  1955.         if (!ASSERT(scriptWrapper, "Couldn't get a script wrapper"))
  1956.             return;
  1957.         if (console.prefs["prettyprint"] && jsdScript.isValid)
  1958.         {
  1959.             displaySourceContext (scriptWrapper.sourceText,
  1960.                                   jsdScript.pcToLine(jsdFrame.pc,
  1961.                                                      PCMAP_PRETTYPRINT),
  1962.                                   sourceContext);
  1963.         }
  1964.         else
  1965.         {
  1966.             displaySourceContext (scriptWrapper.scriptInstance.sourceText,
  1967.                                   jsdFrame.line, sourceContext);
  1968.         }
  1969.     }
  1970. }
  1971.  
  1972. function getFutureBreakpoint (urlPattern, lineNumber)
  1973. {
  1974.     var key = urlPattern + "#" + lineNumber;
  1975.     if (key in console.fbreaks)
  1976.         return console.fbreaks[key];
  1977.     
  1978.     return null;
  1979. }
  1980.  
  1981. function setFutureBreakpoint (urlPattern, lineNumber, props)
  1982. {
  1983.     var key = urlPattern + "#" + lineNumber;
  1984.  
  1985.     if (key in console.fbreaks)
  1986.         return false;
  1987.     
  1988.     var url;
  1989.     
  1990.     for (url in console.scriptManagers)
  1991.     {
  1992.         if (url == urlPattern)
  1993.             console.scriptManagers[url].noteFutureBreakpoint(lineNumber, true);
  1994.     }    
  1995.  
  1996.     for (url in console.files)
  1997.     {
  1998.         if (url == urlPattern)
  1999.             console.files[url].noteFutureBreakpoint(lineNumber, true);
  2000.     }    
  2001.  
  2002.     var fbreak = new FutureBreakpoint (urlPattern, lineNumber);
  2003.     if (props)
  2004.         fbreak.setProperties(props);
  2005.     console.fbreaks[key] = fbreak;
  2006.  
  2007.     dispatch ("hook-fbreak-set", { fbreak: fbreak });
  2008.  
  2009.     return fbreak;
  2010. }
  2011.  
  2012. function clearFutureBreakpoint (urlPattern, lineNumber)
  2013. {
  2014.     var key = urlPattern + "#" + lineNumber;
  2015.     if (!(key in console.fbreaks))
  2016.         return false;
  2017.  
  2018.     var i;
  2019.     var fbreak = console.fbreaks[key];
  2020.     if ("propsWindow" in fbreak)
  2021.         fbreak.propsWindow.close();
  2022.     
  2023.     delete console.fbreaks[key];
  2024.  
  2025.     for (i in fbreak.childrenBP)
  2026.         fbreak.childrenBP[i].parentBP = null;
  2027.  
  2028.     var url;
  2029.     
  2030.     for (url in console.scriptManagers)
  2031.     {
  2032.         if (url.indexOf(urlPattern) != -1)
  2033.             console.scriptManagers[url].noteFutureBreakpoint(lineNumber, false);
  2034.     }
  2035.  
  2036.     for (url in console.files)
  2037.     {
  2038.         if (url == urlPattern)
  2039.             console.files[url].noteFutureBreakpoint(lineNumber, false);
  2040.     }    
  2041.  
  2042.     dispatch ("hook-fbreak-clear", { fbreak: fbreak });
  2043.  
  2044.     return true;
  2045. }
  2046.